![]() |
![]() |
|
Das Ergebnis der Abfrage enthält alle Datensätze der Tabelle authors. Diese Datensätze sind in einer Tabelle enthalten, die durch ein DataTable-Objekt beschrieben wird. 26.5.4 Tabellen- und Spaltenbezeichner zuordnen
|
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
| Dim strSQL As String= "SELECT * FROM authors;" & _ |
| "SELECT * FROM titleauthor;" & _ |
| "SELECT * FROM titles" |
| Dim da As SqlDataAdapter = New SqlDataAdapter(strSQL, con) |
| Dim ds As New DataSet |
| da.Fill(ds) |
| Hinweis |
|
Der SQL Server unterstützt Batch-Abfragen, jedoch nicht unbedingt jede andere Datenbank. Gegebenenfalls müssen Sie sich in der Dokumentation eines Datenbankmanagementsystems die entsprechende Information besorgen. |
Das DataSet beherbergt nun drei Tabellen im lokalen Speicher. In jeder Tabelle sind alle Datensätze der entsprechenden Originaltabelle authors, titles und titleauthor enthalten. Wir könnten nun jeder lokalen Tabelle Datensätze hinzufügen, Datensätze löschen oder ändern. Alle Änderungen werden bei einer erneuten Verbindungsaufnahme unter Aufruf der Update-Methode des DataAdapters in die Originaldatenbank zurückgeschrieben.
Allerdings stehen wir zuvor vor einer Frage: Wie kann ich eine bestimmte Tabelle im DataSet ansprechen, wenn darin mehrere Tabellen enthalten sind? Denn um einen Datensatz zu bearbeiten, muss zuvor festgelegt werden, in welcher Tabelle der entsprechende Datensatz zu finden ist.
Ein DataSet verwaltet alle in ihm enthaltenen Tabellen in einer Auflistung vom Typ Data-TableCollection. Die Referenz auf diese Auflistung liefert die Eigenschaft Tables des DataSet-Objekts.
| Public ReadOnly Property Tables As DataTableCollection |
Der Zugriff auf eine Tabelle im DataSet erfolgt über den Index der Auflistung, dem entweder der Verwaltungsindex der Tabelle oder der Bezeichner der Tabelle übergeben werden kann, z. B.:
ds.Tables(0)
Jetzt sollte man auch noch wissen, dass ein DataTable-Objekt seinen Tabellennamen über die Eigenschaft TableName preisgibt. Mit diesen Kenntnissen können wir jetzt die Namen der Tabellen im DataSet abfragen:
| For Each table As DataTable In ds.Tables |
| Console.WriteLine(table.TableName) |
Die Ausgabe wird nicht – wie vielleicht zu vermuten wäre – authors, titleauthor und titles lauten, sondern:
| Table |
| Table1 |
| Table2 |
Die Zuordnung von Table zu authors, Table1 zu titleauthor und Table2 zu titles ist in den meisten Fällen nicht wünschenswert. Der DataAdapter bietet daher einen Mechanismus, um den Tabellen im Abfrageergebnis einen anderen Namen zuzuordnen: die Eigenschaft TableMappings, welche die Referenz auf ein DataTableMappingCollection-Objekt liefert.
| Public ReadOnly Property TableMappings As DataTableMappingCollection |
In der Auflistung DataTableMappingCollection werden Objekte vom Typ DataTableMapping verwaltet. Jedes dieser Objekte ordnet einer Tabelle im DataSet einen Tabellennamen zu.
Am einfachsten ist es, mit der Add-Methode die Auflistung zu füllen.
| Public Function Add(String, String) As DataTableMapping |
Dazu wird dem ersten Parameter die Zeichenfolge übergeben, unter der die Tabelle per Vorgabe in das DataSet gefüllt wird, dem zweiten Parameter teilt man den gewünschten Tabellennamen mit, unter dem die Tabelle im Code angesprochen wird.
Das folgende Codefragment zeigt, wie Sie die DataTableMappingCollection des DataAdapter-Objekts füllen können. Dabei wird davon ausgegangen, dass die oben angeführte Batch-Abfrage abgesetzt wird. Die Zuordnung muss vor dem Füllen des DataSets mit Fill erfolgen, ansonsten bleibt sie wirkungslos.
| da.TableMappings.Add("Table", "Autoren") |
| da.TableMappings.Add("Table1", "TitelAutor") |
| da.TableMappings.Add("Table2", "Titel") |
| Dim ds As New DataSet |
| da.Fill(ds) |
| ... |
Add ruft implizit den DataTableMapping-Konstruktor auf. Sie können das natürlich auch selbst in die Hand nehmen, müssen dann aber jeder Tabelle über die Eigenschaft SourceTable sagen, welchen Standardnamen sie im DataSet hat, und über DataSetTable, welcher Bezeichner der Tabelle neu zugeordnet werden soll. Das folgende Beispiel zeigt, wie der Code dazu aussieht.
| Dim dtm1 As New DataTableMapping |
| dtm1.SourceTable = "Table" |
| dtm1.DataSetTable = "Autoren" |
| da.TableMappings.Add(dtm1) |
| Dim dtm2 As New DataTableMapping |
| dtm2.SourceTable = "Table1" |
| dtm2.DataSetTable = "TitelAutor" |
| da.TableMappings.Add(dtm2) |
| Dim dtm3 As New DataTableMapping |
| dtm3.SourceTable = "Table2" |
| dtm3.DataSetTable = "Titel" |
| da.TableMappings.Add(dtm3) |
| Dim ds As New DataSet |
| da.Fill(ds) |
| ... |
Die Klasse DataTableMapping gehört zum Namespace System.Data.Common. Deutlich ist zu sehen, dass diese Art der Zuordnung mehr Programmieraufwand bedeutet. Daher steht vermutlich außer Frage, welcher Variante Sie den Vorzug geben.
Jeder Spalte der SELECT-Abfrage wird eine Spalte in der DataTable zugeordnet. Als Spaltenbezeichner verwendet ADO.NET dabei den Spaltennamen der Originaltabelle in der Datenbank. Fragen Sie die Datenquelle mit
| SELECT au_lname, au_fname, city FROM authors |
ab, lauten die Spalten in der DataTable ebenfalls au_lname, au_fname und city. Wünschen Sie andere Spaltenbezeichner, können Sie im SELECT-Statement für die einzelnen Spalten einen Alias angeben, z. B.:
| SELECT au_lname AS Zuname, au_fname AS Vorname, city AS Stadt |
| FROM authors |
Nun würden in der DataTable die Spaltenbezeichner Zuname, Vorname und Stadt lauten.
Sie können alternativ auch einen anderen Mechanismus einsetzen. Ein DataTableMapping-Objekt hat eine eigene Auflistung, mit der den obligatorischen Spaltenbezeichnern neue zugeordnet werden können. Diese Auflistung ist vom Typ DataColumnMappingCollection und enthält DataColumnMapping-Objekte. Jedes dieser Objekte beschreibt für sich eine Neuzuordnung eines Spaltenbezeichners in einer DataTable. Die vielleicht ein wenig komplex anmutenden Zusammenhänge zwischen DataAdapter, DataTableMapping und DataColumnMapping sind in Abbildung 26.8 anschaulich dargestellt.

Hier klicken, um das Bild zu Vergrößern
Abbildung 26.8 Die Hierarchie der Zuordnungsklassen
Die Referenz auf die DataColumnMappingCollection stellt die Eigenschaft ColumnMappings der Klasse DataTableMapping bereit:
| Public ReadOnly Property ColumnMappings _ |
| As DataColumnMappingCollection |
Um eine Neuzuordnung festzulegen, bietet sich auch hier der Weg über die Add-Methode des DataColumnMappingCollection-Objekts an.
| Public Function Add (String, String) As DataColumnMapping |
Analog zur Add-Methode der DataTableMappingCollection wird dem ersten Parameter der ursprüngliche Spaltenbezeichner, dem zweiten Parameter der gewünschte übergeben.
Das folgende Codefragment zeigt den kompletten Code, der notwendig ist, um neben dem Tabellennamen auch die Spaltenbezeichner einer Abfrage neu festzulegen. Zum Schluss werden die Spaltenneuzuordnungen zur Bestätigung an der Konsole ausgegeben. Der Code im Schleifenkopf zur Ausgabe der Spaltenbezeichner dürfte zwar auch ohne weitere Erläuterungen verständlich sein. Wir werden aber dennoch an anderer Stelle darauf zurückkommen.
| Dim strCon As String = "..." |
| Dim con As SqlConnection = New SqlConnection(strCon) |
| Dim strSQL As String = _ |
| "SELECT au_lname, au_fname, city FROM authors" |
| Dim da As SqlDataAdapter = New SqlDataAdapter(strSQL, con) |
| ' Neuzuordnung des Tabellennamens |
| Dim dtm As DataTableMapping = _ |
| da.TableMappings.Add("Table", "Autoren") |
| ' Neuzuordnung der Spaltenbezeichner |
| dtm.ColumnMappings.Add("au_lname", "Zuname") |
| dtm.ColumnMappings.Add("au_fname", "Vorname") |
| dtm.ColumnMappings.Add("city", "Stadt") |
| Dim ds As New DataSet |
| da.Fill(ds) |
| ' Konsolenausgabe der Spaltenbezeichner |
| For Each column As DataColumn In ds.Tables(0).Columns |
| Console.WriteLine(column.ColumnName) |
Übergeben Sie der Fill-Methode anstelle eines DataSet- ein DataTable-Objekt, müssen Sie ein wenig anders vorgehen, um die Spalten mit eigenen Bezeichnern im lokalen Datenspeicher anzusprechen. Dazu erzeugen Sie auch wieder ein DataTableMapping-Objekt, dem Sie die gewünschten Spaltenbezeichner zuordnen. Bei der Instanziierung von DataTable rufen Sie allerdings den parametrisierten Konstruktor auf, dem der im DataTableMappping zugeordnete Tabellenname übergeben wird.
| ... |
| Dim dtm As DataTableMapping = _ |
| da.TableMappings.Add("Table", "Autoren") |
| ' Neuzuordnung der Spaltenbezeichner |
| dtm.ColumnMappings.Add("au_lname", "Zuname") |
| dtm.ColumnMappings.Add("au_fname", "Vorname") |
| dtm.ColumnMappings.Add("city", "Stadt") |
| Dim tbl As DataTable = New DataTable("Autoren") |
| da.Fill(tbl) |
| ... |
Die Neuzuordnung der Tabellen- und Spaltenbezeichner ist eine Option, die vor dem Aufruf der Methode Fill wahrgenommen werden kann oder nicht. Der DataAdapter prüft vor dem Füllen des DataSets, ob die Zuordnungsauflistungen gefüllt sind. Dabei interessiert er sich besonders für die Spaltenzuordnungen.
Für jede Spalte des Abfrageergebnisses überprüft der DataAdapter zuerst, ob dafür eine Zuordnung in der DataColumnMappingCollection angegeben ist. Existiert eine solche nicht, überprüft er im nächsten Schritt seine MissingMappingAction-Eigenschaft. Hier findet er die Antwort darauf, wie er mit einer fehlenden Spaltenangabe umzugehen hat.
Standardmäßig werden Spalten, die nicht im DataColumnMapping-Objekt angegeben sind, mit dem Namen, den sie in der Originaltabelle haben, in die entsprechende DataTable eingetragen. Der DataAdapter kann aber auch angewiesen werden, alle Spalten, die nicht in der Zuordnungstabelle enthalten sind, zu ignorieren. Eine dritte Möglichkeit ist, eine Ausnahme auszulösen, wenn keine Zuordnung angegeben ist.
Die Eigenschaft MissingMappingAction teilt dem DataAdapter mit, wie er zu verfahren hat. Diese Eigenschaft ist vom Typ der gleichnamigen Enumeration MissingMappingAction.
| Public Overridable Property MissingMappingAction As MissingMappingAction |
Die drei Member der Enumeration lauten Error, Ignore und Passthrough. Letztere ist die Standardeinstellung.
| Member | Beschreibung |
| Error | Fehlt eine Spaltenzuordnung, wird eine Ausnahme ausgelöst. |
| Ignore | Fehlt eine Spaltenzuordnung, wird die Spalte in der DataTable ignoriert. |
| Passthrough | Fehlt eine Spaltenzuordnung, wird die Spalte unter ihrem ursprünglichen Namen der DataTable hinzugefügt. |
| << zurück |
|
||||||||||||||
|
||||||||||||||
|
||||||||||||||
|
||||||||||||||
Copyright © Galileo Press 2007
Für Ihren privaten Gebrauch dürfen Sie die Online-Version natürlich ausdrucken.
Ansonsten unterliegt das <openbook> denselben Bestimmungen, wie die
gebundene Ausgabe: Das Werk einschließlich aller seiner Teile ist urheberrechtlich
geschützt. Alle Rechte vorbehalten einschließlich der Vervielfältigung, Übersetzung,
Mikroverfilmung sowie Einspeicherung und Verarbeitung in elektronischen Systemen.